package com.xiaotu.lambda.java2;

import org.junit.Test;

import javax.lang.model.element.VariableElement;
import java.io.PrintStream;
import java.util.Comparator;
import java.util.function.BiPredicate;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Supplier;

/**
 * 方法引用的使用
 * <p>
 * 1.使用情境：当要传递给Lambda体的操作，已经有实现的方法了，可以使用方法引用！
 * 2.方法引用，本质上就是Lambda表达式，而Lambda表达式作为函数式接口的实例。所以
 * 方法引用，也是函数式接口的实例。
 * <p>
 * 3.使用格式：类（或对象）:: 方法名
 * 4.具体分为如下的三种情况：
 * 情况1      对象 :: 非静态方法
 * 情况2      类 :: 静态方法
 * <p>
 * 情况3      类 :: 非静态方法
 * <p>
 * 5.方法引用使用的要求：要求接口中的抽象方法的形参列表和返回值类型与
 * 方法引用的方法的形参列表和返回值类型！（针对于情况1和情况2）
 *
 * @author 张晓
 * @create 2021-06-02 17:48
 */
public class MethodRefTest {

    //情况一：对象::实例方法
    //Consumer中的void accept (T t)
    //PrintStream中的void println(T t)
    @Test
    public void test1() {
        Consumer<String> con1 = s -> System.out.println(s);
        con1.accept("北京");

        System.out.println("*********************");

        PrintStream ps = System.out;
        Consumer<String> con2 = ps::println;
        con2.accept("上海");
    }

    //Supplier中的T get()
    //Employee中的String getName()
    @Test
    public void test2() {
        Employee emp = new Employee(1001, "Tom", 23, 5600);
        Supplier<String> sup1 = () -> emp.getName();
        System.out.println(sup1.get());

        System.out.println("*********************");

        Supplier<String> sup2 = emp::getName;
        System.out.println(sup2.get());
    }

    //情况二：类::静态方法
    //Comparator中的int compare(T t1,T t2)
    //Integer中的int compare(T t1,T t2)
    @Test
    public void test3() {
        Comparator<Integer> com1 = (t1, t2) -> Integer.compare(t1, t2);
        System.out.println(com1.compare(11, 12));

        System.out.println("*********************");

        Comparator<Integer> com2 = Integer::compare;
        System.out.println(com2.compare(12, 9));
    }

    //Function中的R apply(T t)
    //Math 中的Long round(Double d)
    @Test
    public void test4() {
        Function<Double, Long> func = new Function<Double, Long>() {
            @Override
            public Long apply(Double d) {
                return Math.round(d);
            }
        };
        System.out.println("**********************");

        Function<Double, Long> func1 = d -> Math.round(d);
        System.out.println(func1.apply(12.1));
        System.out.println("**********************");

        Function<Double, Long> func2 = Math::round;
        System.out.println(func2.apply(12.6));
    }

    //情况三：类::实例方法 (有难度)
    //Comparator中的int compare(T t1,T t2)
    //String 中的int t1.compareTo(t2)
    @Test
    public void test5() {
        Comparator<String> com1 = (s1, s2) -> s1.compareTo(s2);
        System.out.println(com1.compare("abc", "abd"));
        System.out.println("**********************");
        Comparator<String> com2 = String::compareTo;
        System.out.println(com2.compare("abc", "abm"));
    }

    //BiPredicate中的boolean test(T t1,T t2)
    //String 中的boolean t1.equals(t2)
    @Test
    public void test6() {
        BiPredicate<String, String> pre1 = (s1, s2) -> s1.equals(s2);
        System.out.println(pre1.test("abc", "abc"));
        System.out.println("**********************");
        BiPredicate<String, String> pre2 = String::equals;
        System.out.println(pre2.test("abc", "abc"));
    }

    //Function 中的R apply(T t)
    //Employee 中的String getName()
    @Test
    public void test7() {
        Employee employee = new Employee(1001, "Jerry", 23, 6000);
        Function<Employee, String> func1 = e -> e.getName();
        System.out.println(func1.apply(employee));
        System.out.println("**********************");
        Function<Employee, String> func2 = Employee::getName;
        System.out.println(func2.apply(employee));
    }

}
